# Adding Custom IP to the System

#### Introduction

This lab guides you through the process of creating and adding a custom peripheral to a processor system by using the Vivado IP Packager. You will create an AXI4Lite interface peripheral.

### **Objectives**

After completing this lab, you will be able to:

- Use the IP Packager feature of Vivado to create a custom peripheral
- · Modify the functionality of the IP
- · Add the custom peripheral to your design
- Add pin location constraints
- Add block memory to the system

#### **Procedure**

This lab is separated into steps that consist of general overview statements that provide information on the detailed instructions that follow. Follow these detailed instructions to progress through the lab.

This lab comprises 4 primary steps: You will use a peripheral template to create a peripheral, Package the IP using IP Packager, import, add and connect the IP in the design, and add the Block RAM (BRAM) Memory.

### **Design Description**

You will extend the Lab 2 hardware design by creating and adding an AXI peripheral (refer to LED\_IP in **Figure 1**) to the system, and connecting it to the LEDs on the Zynq board you are using. You will use the IP Packager to generate the custom IP. Next, you will connect the peripheral to the system and add pin location constraints to connect the LED display controller peripheral to the on-board LED display. Finally, you will add BRAM Controller and BRAM before generating the bitstream.



Figure 1. Design Updated from Previous Lab



#### General Flow for this Lab



In the instructions below;

{sources} refers to: C:\xup\embedded\2015 2 zyng sources

{ labs } refers to : C:\xup\embedded\2015\_2\_zynq\_labs

{labsolutions} for the ZedBoard refers to: C:\xup\embedded\2015\_2\_zedboard\_labsolution or for the Zybo refers to: C:\xup\embedded\2015\_2\_zedboard\_labsolution

### Create a Custom IP using the Create and Package IP Wizard Step 1

- 1-1. Use the provided axi\_lite slave peripheral template and the custom IP source code to create a custom IP.
- 1-1-1. Open Vivado by selecting Start > All Programs > Xilinx Design Tools > Vivado 2018.3 > Vivado 2018.3
- 1-1-2. Click Manage IP and select New IP Location and click Next in the New IP Location window
- **1-1-3.** Select **Verilog** as the *Target Language*, **Mixed** as the *Simulator language*, and for *IP location*, type *{labs}/led\_ip* and click **Finish** (leave other settings as defaults and click **OK** if prompted to create the directory)





#### Figure 2. New IP Location form

A Virtex 7 part is chosen for this project, but later compatibility for other devices will be added to the packaged IP.



Click OK for the above prompt. You will get to the following display.



### 1-2. Run the Create and Package IP Wizard

1-2-1. Select Tools > Create and Package New IP





1-2-2. In the window, click Next



- 1-2-3. Select Create a new AXI4 peripheral, and click Next
- 1-2-4. Fill in the details for the IP

Name: led\_ip

Display Name: led\_ip\_v1.0

(Fill in a description, Vendor Name, and URL)





Figure 3. Updating Peripheral Details form

- 1-2-5. Click Next
- 1-2-6. Change the Name of the interface to S\_AXI
- 1-2-7. Leave the other settings as default and click Next (Lite interface, Slave mode, Data Width 32, Registers 4)



Figure 4. Naming the AXI interface

**1-2-8.** Select *Edit IP* and click **Finish** (a new Vivado Project will open)







#### 1-3. Create an interface to the LEDs

**1-3-1.** In the sources panel, double-click the **led\_ip\_v1\_0.v** file.

This file contains the HDL code for the interface(s) selected above. The top level file contains a module which implements the AXI interfacing logic, and an example design to write to and read from the number of registers specified above. This template can be used as a basis for creating custom IP. A new parameterized output port to the LEDs will be created at the top level of the design, and the AXI write data in the sub-module will be connected back up to the external LED port.

Scroll down to line 7 where a user *parameters* space is provided.

#### 1-3-2. Add the line:



#### parameter integer LED\_WIDTH = 8,

**1-3-3.** Go to line 18 and add the line:

#### output wire [LED\_WIDTH-1:0] LED,

(Notice the extra comma needed at the end of each line)

```
module led ip v1 0 #
 5
 6
          // Users to add parameters here
 7
          parameter integer LED WIDTH
                                                  = 8,
8
          // User parameters ends
9
          // Do not modify the parameters beyond this line
10
11
12
          // Parameters of Axi Slave Bus Interface S AXI
13
          parameter integer C_S_AXI_DATA_WIDTH
14
          parameter integer C_S_AXI_ADDR_WIDTH
15
      )
16
      (
17
          // Users to add ports here
          output wire [LED_WIDTH-1:0]
18
                                           LED,
19
          // User ports ends
```

Figure 5. Adding users parameter, and port definition

**1-3-4.** Insert the following at line ~48:

#### .LED\_WIDTH(LED\_WIDTH),

**1-3-5.** Insert the following at line ~52:

#### .LED(LED),

```
46 // Instantiation of Axi Bus Interface S AXI
47
      led ip v1 0 S AXI # (
48
          .LED_WIDTH(LED_WIDTH),
49
          .C S AXI DATA WIDTH (C S AXI DATA WIDTH),
50
          .C S AXI ADDR WIDTH (C S AXI ADDR WIDTH)
51
      ) led ip v1 0 S AXI inst (
52
          .LED (LED),
53
          .S AXI ACLK(s axi aclk),
54
          .S_AXI_ARESETN(s_axi_aresetn),
```

Figure 6. Adding port connection with a lower-level module

- 1-3-6. Save the file by selecting File > Save File
- 1-3-7. Expand led\_ip\_v1\_0 in the sources view if necessary, and open led\_ip\_v1\_0\_S\_AXI.v



1-3-8. Add the LED parameter and port to this file too, at lines 7 and 18

```
4
      module led ip v1 0 S AXI #
 5
 6
          // Users to add parameters here,
          parameter integer LED WIDTH
 7
                                                   = 8,
 8
          // User parameters ends
 9
          // Do not modify the parameters beyond this line
10
11
          // Width of S AXI data bus
12
          parameter integer C S AXI DATA WIDTH
                                                   = 32,
13
          // Width of S AXI address bus
14
          parameter integer C_S_AXI_ADDR_WIDTH
15
      )
16
17
          // Users to add ports here
18
          output wire [LED WIDTH-1:0]
                                            LED,
19
          // User ports ends
```

Figure 7. Declaring users port in the lower-level module for the Zybo

1-3-9. Scroll down to ~line 400 and insert the following code to instantiate the user logic for the LED IP

(This code can be typed directly, or copied from the user\_logic\_instantiation.txt file in the lab3 source folder.)

```
399
       // Add user logic here
400
       lab3_user_logic # (
401
          .LED_WIDTH(LED_WIDTH)
402
403
       U1 (
404
          .S_AXI_ACLK(S_AXI_ACLK),
          .slv_reg_wren(slv_reg_wren),
406
          .axi awaddr(axi awaddr[C S AXI ADDR WIDTH-1:ADDR LSB]),
407
          .S AXI WDATA(S AXI WDATA),
          .S AXI ARESETN(S AXI ARESETN),
408
409
          .LED(LED)
410
       );
411
       // User logic ends
```

Figure 8. Instantiating lower-level user module

Check all the signals that are being connected and where they originate.

- 1-3-10. Save the file by selecting File > Save File
- 1-3-11. Click on the Add Sources in the Flow Navigator pane, select Add or Create Design Sources, click Next, then click the + icon then Add Files..., or click the "Add Files" button, download lab3\_user\_logic.v file from beachboard dropbox to your current project directory, select it and click OK, and then click Finish to add the file.













Check the contents of this file to understand the logic that is being implemented. Notice the formed hierarchy.



**1-3-12.** Click **Run Synthesis** and *Save* if prompted. (This is to check the design synthesizes correctly before packaging the IP. If this was your own design, you would simulate it and verify functionality before proceeding)





#### 1-3-13. Check the Messages tab for any errors and correct if necessary before moving to the next step

When Synthesis completes successfully, click Cancel.



#### 1-4. Package the IP

#### 1-4-1. Click on the Package IP - led\_ip tab



Figure 9. Package IP



- **1-4-2.** For the IP to appear in the IP catalog in particular categories, the IP must be configured to be part of those categories. To change which categories the IP will appear in the IP catalog click + icon in the *Categories* section. This opens the Choose IP Categories window
- **1-4-3.** For the purpose of this exercise, uncheck the **AXI Peripheral** box and check the **Basic Elements** and click **OK**.



Figure 10. Specify the category for IP Packager IP



- 1-4-4. Select Compatibility. This shows the different Xilinx FPGA Families that the IP supports. The value is inherited from the device selected for the project.
- 1-4-5. Click the + icon then Add Family Explicitly... from the menu.
- **1-4-6.** Select the **Zynq** family as we will be using this IP on the Zybo, and click **OK**.





1-4-7. You can also customize the address space and add memory address space using the Addressing and Memory category. We won't make any changes.



#### 1-4-8. Click on File Groups and click Merge changes from File Groups Wizard



Figure 11. Updating the file group - before updating

This is to update the IP Packager with the changes that were made to the IP and the *lab3\_user\_logic\_\*.v* file that was added to the project.

1-4-9. Expand Verilog Synthesis and notice lab3\_user\_logic.v has been included



**1-4-10.** Click on **Customization Parameters** and again *Merge changes from Customization Parameters Wizard* 





Notice that the Ports and Interfaces view now shows the user created LED port



Figure 12. User parameter and port

1-4-11. Select Customization Parameters, expand Hidden Parameters, right-click on LED\_WIDTH, and select Import IP Parameters... and click OK.





After click OK on the above window, you will see the following display:



1-4-12. Select Customization GUI and notice that the Led Width is visible.



Figure 13. User customization parameter



1-4-13. Select Review and Package, and notice the path where the IP will be created.



1-4-14. Click Package IP or Re-Package IP. Click Yes and the project will close when complete.



1-4-15. In the original Vivado window click File > Close Project

## **Modify the Project Settings**

Step 3

- 2-1. Open the previous project: lab2 project and save the project as lab3. Set Project Settings to point to the created IP repository.
- **2-1-1.** Start the Vivado if necessary and open either the lab2 project you created in the previous lab or the lab2 project in the labsolution directory
- **2-1-2.** Select **File > Save Project As ...** to open the *Save Project As* dialog box. Enter **lab3** as the project name. Make sure that the *Create Project Subdirectory* option is checked, the project directory path is {labs}\ and click **OK**.

This will create the lab3 directory and save the project and associated directory with lab3 name.



- **2-1-3.** Click **Project Settings** in the *Flow Navigator* pane.
- 2-1-4. Select IP in the left pane of the Project Settings form. Expand IP and select Repository.



2-1-5. Click on the + button, browse to {labs}\led\_ip and click Select.

The led\_ip\_v1.0 IP will appear the **IP in the Selected Repository** window.







Figure 14. Specify IP Repository

2-1-6. Click OK twice.

### Add the Custom IP, BRAM, and the Constraints

Step 4

- 3-1. Add led\_ip to the design and connect to the AXI4Lite interconnect in the IPI. Make internal and external port connections. Establish the LED port as external FPGA pins.
- 3-1-1. Click Open Block Design under IP Integrator in the Flow Navigator pane
- **3-1-2.** Click the Add IP icon **+** and search for **led\_ip\_v1.0** in the catalog by typing "led" in the search field.



Figure 15. Searching for led\_ip in the IP Catalog

**3-1-3.** Press Enter to add the core to the design.





**3-1-4.** Right click the IP instance, select **Block Properties** on the popup window, Select the IP in the block diagram and change the instance name to **led\_ip** in the **Block Properties** view.



**3-1-5.** Double click the block to open the **Re-customize IP** view.





**3-1-6.** Set the width to 4 for Zybo board.



Figure 16. Configure the LED IP LED\_WIDTH

- 3-1-7. Click OK.
- **3-1-8.** Click on **Run Connection Automation**, select /**Ied\_ip/S\_AXI** and click **OK** to automatically make the connection from the AXI Interconnect to the IP.





Click the **Regenerate Layout** button to redraw the diagram.





**3-1-9.** Select the *LED* port on the led\_ip instance (by clicking on its pin), right-click and select **Make External**.



Figure 17. LED external port added and connected

**3-1-10.** Select the **Address Editor** tab and verify that an address has been assigned to *led\_ip*.



Figure 18. Address assigned for led\_ip

### 3-2. Add BRAM to the design

- **3-2-1.** In the Block Diagram, click the Add IP icon + and search for BRAM and add one instance of the **AXI BRAM Controller**
- **3-2-2.** Run *Connection Automation*, select **axi\_bram\_ctrl\_0/S\_AXI** on the **Run Connection Automation** view, and click **OK** when prompted to connect it to the **M\_AXI\_GP0** Master.







3-2-3. Double click on the block to customize it and change the number of BRAM interfaces to 1 and click OK.



Notice that the AXI Protocol being used is AXI4 instead of AXI4Lite since BRAM can provide higher bandwidth and the controller can support burst transactions.



Figure 19. Customize BRAM controller





3-2-4. Click on Run Connection Automation to add and connect a Block Memory Generator by selecting axi\_bram\_ctrl\_0/BRAM\_PORTA and click OK (This could be added manually)



3-2-5. Validate the design to ensure there are no errors (F6), and click the Regenerate Layout button ( C) to redraw the diagram.



The design should look similar to the figure below.



### Rename LED\_0[3:0] port to LED[3:0].



Figure 20. Completed Block Diagram

**3-2-6.** In the Address editor, notice the Range of the axi\_bram\_ctrl\_0 is 8K. We will leave it at that.





Figure 21. Adjusting memory size

- **3-2-7.** Press **F6** to validate the design one last time.
- 3-3. Add the provided lab3\_\*.xdc constraints file.
- 3-3-1. Click Add Sources in the Flow Navigator pane, select Add or Create Constraints, and click Next.



3-3-2. Click the + button, and then Add Files.... Download lab3\_zybo.xdc from beachboard dropbox to your Lab3 folder, browse to the folder, select lab3\_zybo.xdc for the Zybo.



- 3-3-3. Click Finish to add the file.
- **3-3-4.** Expand Constraints folder in the *Sources* pane, and double click the **lab3\_zybo.xdc** file entry to see its content. This file contains the pin locations and IO standards for the LEDs on the Zynq board. This information can usually be found in the manufacturer's datasheet for the board.



**3-3-5.** Right click on *design\_1.bd* and select *Generate output products*, click "Generate" on the next prompt.





**3-3-6.** Click on **Generate Bitstream** and click **Yes** if prompted to save the Block Diagram, and click **Yes** again if prompted to launch synthesis and implementation. Click **Cancel** when prompted to *Open the Implemented Design* 









### Conclusion

Vivado IP packager was used to import a custom IP block into the IP library. The IP block was then added to the system. Connection automation was run where available to speed up the design of the system by allowing Vivado to automatically make connections between IP. An additional BRAM was added to the design. Finally, pin location constraints were added to the design.

